| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194 |
- import { getServerSideConfig } from "@/app/config/server";
- import {
- ANTHROPIC_BASE_URL,
- Anthropic,
- ApiPath,
- DEFAULT_MODELS,
- ServiceProvider,
- ModelProvider,
- } from "@/app/constant";
- import { prettyObject } from "@/app/utils/format";
- import { NextRequest, NextResponse } from "next/server";
- import { auth } from "../../auth";
- import { isModelAvailableInServer } from "@/app/utils/model";
- import { cloudflareAIGatewayUrl } from "@/app/utils/cloudflare";
- const ALLOWD_PATH = new Set([Anthropic.ChatPath, Anthropic.ChatPath1]);
- async function handle(
- req: NextRequest,
- { params }: { params: { path: string[] } },
- ) {
- console.log("[Anthropic Route] params ", params);
- if (req.method === "OPTIONS") {
- return NextResponse.json({ body: "OK" }, { status: 200 });
- }
- const subpath = params.path.join("/");
- if (!ALLOWD_PATH.has(subpath)) {
- console.log("[Anthropic Route] forbidden path ", subpath);
- return NextResponse.json(
- {
- error: true,
- msg: "you are not allowed to request " + subpath,
- },
- {
- status: 403,
- },
- );
- }
- const authResult = auth(req, ModelProvider.Claude);
- if (authResult.error) {
- return NextResponse.json(authResult, {
- status: 401,
- });
- }
- try {
- const response = await request(req);
- return response;
- } catch (e) {
- console.error("[Anthropic] ", e);
- return NextResponse.json(prettyObject(e));
- }
- }
- export const GET = handle;
- export const POST = handle;
- export const runtime = "edge";
- export const preferredRegion = [
- "arn1",
- "bom1",
- "cdg1",
- "cle1",
- "cpt1",
- "dub1",
- "fra1",
- "gru1",
- "hnd1",
- "iad1",
- "icn1",
- "kix1",
- "lhr1",
- "pdx1",
- "sfo1",
- "sin1",
- "syd1",
- ];
- const serverConfig = getServerSideConfig();
- async function request(req: NextRequest) {
- const controller = new AbortController();
- let authHeaderName = "x-api-key";
- let authValue =
- req.headers.get(authHeaderName) ||
- req.headers.get("Authorization")?.replaceAll("Bearer ", "").trim() ||
- serverConfig.anthropicApiKey ||
- "";
- let path = `${req.nextUrl.pathname}`.replaceAll(ApiPath.Anthropic, "");
- let baseUrl =
- serverConfig.anthropicUrl || serverConfig.baseUrl || ANTHROPIC_BASE_URL;
- if (!baseUrl.startsWith("http")) {
- baseUrl = `https://${baseUrl}`;
- }
- if (baseUrl.endsWith("/")) {
- baseUrl = baseUrl.slice(0, -1);
- }
- console.log("[Proxy] ", path);
- console.log("[Base Url]", baseUrl);
- const timeoutId = setTimeout(
- () => {
- controller.abort();
- },
- 10 * 60 * 1000,
- );
- // try rebuild url, when using cloudflare ai gateway in server
- const fetchUrl = cloudflareAIGatewayUrl(`${baseUrl}${path}`);
- const fetchOptions: RequestInit = {
- headers: {
- "Content-Type": "application/json",
- "Cache-Control": "no-store",
- [authHeaderName]: authValue,
- "anthropic-version":
- req.headers.get("anthropic-version") ||
- serverConfig.anthropicApiVersion ||
- Anthropic.Vision,
- },
- method: req.method,
- body: req.body,
- redirect: "manual",
- // @ts-ignore
- duplex: "half",
- signal: controller.signal,
- };
- // #1815 try to refuse some request to some models
- if (serverConfig.customModels && req.body) {
- try {
- const clonedBody = await req.text();
- fetchOptions.body = clonedBody;
- const jsonBody = JSON.parse(clonedBody) as { model?: string };
- // not undefined and is false
- if (
- isModelAvailableInServer(
- serverConfig.customModels,
- jsonBody?.model as string,
- ServiceProvider.Anthropic as string,
- )
- ) {
- return NextResponse.json(
- {
- error: true,
- message: `you are not allowed to use ${jsonBody?.model} model`,
- },
- {
- status: 403,
- },
- );
- }
- } catch (e) {
- console.error(`[Anthropic] filter`, e);
- }
- }
- // console.log("[Anthropic request]", fetchOptions.headers, req.method);
- try {
- const res = await fetch(fetchUrl, fetchOptions);
- // console.log(
- // "[Anthropic response]",
- // res.status,
- // " ",
- // res.headers,
- // res.url,
- // );
- // to prevent browser prompt for credentials
- const newHeaders = new Headers(res.headers);
- newHeaders.delete("www-authenticate");
- // to disable nginx buffering
- newHeaders.set("X-Accel-Buffering", "no");
- return new Response(res.body, {
- status: res.status,
- statusText: res.statusText,
- headers: newHeaders,
- });
- } finally {
- clearTimeout(timeoutId);
- }
- }
|